PlutoTest.jl
This notebook introduces visual testing:
3.0
466
8
1
2
3
4
5
6
7
Tests have time-travel functionality built in! Click on the tests above.
Time travel
This notebook contains visual debugging:
UndefVarError: plot not defined
Here is what happened, the most recent locations are first:
(You need Pluto#main
to run this notebook)
1
3
1
3
1
4
missing
4
4
0.0902303
0.735005
0.536793
0.654047
0.170118
0.895947
0.657815
0.705666
0.997585
0.158081
0.918667
0.308976
0.125568
0.159368
0.493538
0.994795
0.146627
0.643642
0.547981
0.194379
0.0260783
0.647428
0.0302074
0.497938
0.826832
0.0956063
0.036913
0.84084
0.76434
0.867522
0.636591
0.682675
0.246418
0.891337
0.0838079
0.798857
0.636498
0.770244
0.452577
0.901059
2
0.228512
0.255628
0.654674
0.601017
true
true
4
4
1
8
true
true
true
true
true
true
true
true
false
false
1.0
1.41421
4.47214
5.0
6.0
7.0
8.0
9.0
0.877185
0.577285
0.275266
0.276778
0.475851
0.557806
0.117438
0.0478049
0.882613
0.45402
0.688345
0.230317
0.931502
0.152388
0.840507
0.394254
0.654462
0.807972
0.432503
0.655838
0.727712
0.634316
0.668238
0.126186
0.184151
0.651291
0.220209
0.155677
0.511714
0.940578
0.303566
0.238192
0.851021
0.597958
0.746384
0.419846
0.682042
0.5853
0.159401
0.176118
0.699317
0.457702
0.267425
0.695037
0.127151
0.0753123
0.634633
0.729485
0.806144
0.927861
0.0985106
0.318337
0.154277
0.0458107
0.260618
0.525212
0.651604
0.613178
0.903134
0.636669
always_false (generic function with 1 method)
"pt-dot {\n\tflex: 0 0 auto;\n\tbackground: grey;\n\twidth: 1em;\n\theight: 1em;\n\tbottom: -.1em;\n\tborder-radius: 100%;\n\tmargin-right: .7em;\n\tdisplay: block;\n\tposition: relative;\n\tcursor: pointer;\n}\n\npt-dot.floating {\n\tposition: fixed;\n\tz-index: 60;\n\tvisibility: hidden;\n\ttransition: transform linear 120ms;\n" ⋯ 1338 bytes ⋯ "-frame-viewer {\n max-width: 100%;\n}\n.pluto-test.expanded > p-frame-viewer > p-frames > slotted-code > line-like {\n\tflex-wrap: wrap;\n}\n.pluto-test.expanded > p-frame-viewer > p-frames > slotted-code > line-like > pluto-display[mime=\"application/vnd.pluto.tree+object\"] {\n\t/*flex-basis: 100%;*/\n}\n"
"p-frame-viewer {\n\tdisplay: inline-flex;\n\tflex-direction: column;\n}\np-frames,\np-frame-controls {\n\tdisplay: inline-flex;\n}\n"
@test_deprecated (macro with 1 method)
Type definitions
Any
Test macro
2
4
6
8
10
12
14
16
18
20
22
24
26
28
30
step_by_step (generic function with 1 method)
test (generic function with 1 method)
begin
var"#715#expr_raw" = $(QuoteNode(:(x == [1, 2 + i])))
try
var"#716#steps" = begin
Main.workspace#6.Any[$(QuoteNode(:(x == [1, 2 + i]))), (Main.workspace#3.onestep_light)($(Expr(:copyast, :($(QuoteNode(:(x == [1, 2 + i])))))); m = Main.workspace#7)...]
end
var"#717#result" = Main.workspace#6.unwrap_computed(Main.workspace#6.last(var"#716#steps"))
if var"#717#result" === true
Main.workspace#6.CorrectCall(var"#715#expr_raw", var"#716#steps")
else
Main.workspace#6.WrongCall(var"#715#expr_raw", var"#716#steps")
end
catch var"#720#e"
Main.workspace#6.rethrow(var"#720#e")
end
end
0.177692
0.0299733
0.199534
0.5229
0.549704
0.966359
0.652341
0.98215
0.787322
0.0531975
0.517717
0.580231
0.0221529
0.157678
0.363806
0.441299
0.013177
0.628578
0.00273435
0.183582
0.123697
0.0161145
0.311343
0.275197
0.486211
0.0224361
0.678502
0.219122
0.76761
0.231894
0.130068
0.679648
0.777732
0.461474
0.0565795
0.37392
0.761637
0.988155
0.276853
0.384182
-4
0.745117
0.357617
0.588769
0.592227
0.872223
0.66986
0.152274
0.815722
0.606807
0.393058
0.979069
0.79169
0.246989
0.267678
0.154557
0.422457
0.29737
0.155406
0.43116
0.682777
0.750607
0.579246
0.19258
0.080015
0.83496
0.558147
0.315809
0.674432
0.569012
0.424698
false
"asdasd"
"asd asd"
"asdasd"
"asdasd"
"asdasd"
"asdasd"
"asdasd"
"asdasd"
"asd"
"asd asd"
"asd"
"asdasd"
"asd"
"asdasd"
"asd"
"asdasd"
"asd"
"asdasd"
"asd"
"asdasd"
"asd"
true
embed_display (generic function with 1 method)
flatmap (generic function with 1 method)
embed_display (generic function with 1 method)
div (generic function with 1 method)
div (generic function with 2 methods)
"x + 1"
expr_to_str (generic function with 2 methods)
prettycolors (generic function with 1 method)
remove_linenums (generic function with 1 method)
remove_linenums (generic function with 2 methods)
UInt64
expr_hash (generic function with 1 method)
expr_hash (generic function with 2 methods)
Time travel evaluation
In Julia, expressions are objects! This means that, before evaluation, code is expressed as a Julia object:
:(first([56, sqrt(9)]))
Expr
head: Symbol call
args: Array{Any}((2,))
1: Symbol first
2: Expr
head: Symbol vect
args: Array{Any}((2,))
1: Int64 56
2: Expr
head: Symbol call
args: Array{Any}((2,))
1: Symbol sqrt
2: Int64 9
You can use Core.eval
to evaluate expressions at runtime:
56.0
But did you know that you can also partially evaluate expressions?
:([56, sqrt(9)])
56.0
3.0
:(first([56.0, 3.0]))
Expr
head: Symbol call
args: Array{Any}((2,))
1: Symbol first
2: Array{Float64}((2,)) [56.0, 3.0]
Here, ex2
is not a raw Expr
— it contains an evaluated array!
Computed
struct
Our time travel mechanism will be based on the partial evaluation principle introduced above. To differentiate between computed results and the original expression, we will wrap all computed results in a struct
.
:(first(Computed([56.0, 3.0])))
Expr
head: Symbol call
args: Array{Any}((2,))
1: Symbol first
2: Main.workspace#3.Computed
x: Array{Float64}((2,)) [56.0, 3.0]
We also add a function to recursively unwrap an expression with Computed
entries:
unwrap_computed (generic function with 1 method)
unwrap_computed (generic function with 2 methods)
unwrap_computed (generic function with 3 methods)
:(first([56.0, 3.0]))
Stepping function
begin
sqrt(Computed(2.0)) + 2
end
begin
Computed(1.4142135623730951) + 2
end
begin
Computed(3.414213562373095)
end
Computed(3.414213562373095)
onestep_light (generic function with 1 method)
onestep_light (generic function with 2 methods)
@eval_step_by_step (macro with 1 method)
quote
#= /home/runner/work/disorganised-mess/disorganised-mess/testing and debugging 3.jl#==#e1c306e3-0a47-4149-a9fb-ec7ab380fa11:6 =#
Main.workspace#3.Any[$(QuoteNode(:(x == [1, 2]))), (Main.workspace#3.onestep_light)($(Expr(:copyast, :($(QuoteNode(:(x == [1, 2])))))); m = Main.workspace#4)...]
end
:(sqrt(sqrt(length([1, 2]))))
:(sqrt(sqrt(length(Computed([1, 2])))))
:(sqrt(sqrt(Computed(2))))
:(sqrt(Computed(1.41421)))
1.18921
:(xasdf = 123)
123
UndefVarError: xasdf not defined
Here is what happened, the most recent locations are first:
begin
Computed(3)
2 + 3
4 + 5
sqrt(sqrt(sqrt(5)))
end
begin
Computed(3)
Computed(5)
4 + 5
sqrt(sqrt(sqrt(5)))
end
begin
Computed(3)
Computed(5)
Computed(9)
sqrt(sqrt(sqrt(5)))
end
begin
Computed(3)
Computed(5)
Computed(9)
sqrt(sqrt(Computed(2.23606797749979)))
end
begin
Computed(3)
Computed(5)
Computed(9)
sqrt(Computed(1.4953487812212205))
end
begin
Computed(3)
Computed(5)
Computed(9)
Computed(1.2228445449938519)
end
Computed(1.2228445449938519)
can_interpret (generic function with 1 method)
can_interpret_call_arg (generic function with 1 method)
can_interpret_call_arg (generic function with 2 methods)
true
can_interpret (generic function with 2 methods)
Displaying objects inside code
Slotting
We walk through the expression tree. Whenever we find a Computed
object, we generate a random key (e.g. iosjddfo
), we add it to our dictionary (found
). In the expression, we replace the Computed
object with a placeholder symbol __slotiosjddfo__
. We will later be able to match the object to this slot.
slot! (generic function with 1 method)
slot! (generic function with 2 methods)
slot! (generic function with 3 methods)
slot (generic function with 1 method)
:__slotpjashjuaayxeuasy__
3
quote __slotpjashjuaayxeuasy__ + (7 - 6) 2 + 3 4 + 5 sqrt(sqrt(sqrt(5))) end
:__slotxttealnvgdxraxbc__
1
:__slotggigdjoatdpfffnc__
3
quote __slotggigdjoatdpfffnc__ + __slotxttealnvgdxraxbc__ 2 + 3 4 + 5 sqrt(sqrt(sqrt(5))) end
:__slotacvtpkbtsjduxabs__
4
quote __slotacvtpkbtsjduxabs__ 2 + 3 4 + 5 sqrt(sqrt(sqrt(5))) end
:__slotdckaushthmqaqmtb__
5
:__slotsuwejrfdjjpubroz__
4
quote __slotsuwejrfdjjpubroz__ __slotdckaushthmqaqmtb__ 4 + 5 sqrt(sqrt(sqrt(5))) end
:__slotsibqusrxknbzxnzt__
5
:__slotndvzvicmaogqpafz__
4
:__slottwpxonfelhdzlkkt__
9
quote __slotndvzvicmaogqpafz__ __slotsibqusrxknbzxnzt__ __slottwpxonfelhdzlkkt__ sqrt(sqrt(sqrt(5))) end
:__slotmkgnmedmufjkequb__
5
:__slotmshcqpkylqlihurt__
4
:__slottcycxiflssxynaeg__
9
:__slotmccabjwojqbcwwgg__
2.23607
quote __slotmshcqpkylqlihurt__ __slotmkgnmedmufjkequb__ __slottcycxiflssxynaeg__ sqrt(sqrt(__slotmccabjwojqbcwwgg__)) end
:__slotxjuovedfnqfthnsd__
5
:__slotoilrzohvsemqjolq__
9
:__slotmpryfmjfndhxqkit__
4
:__slotaafdonjubicrduvj__
1.49535
quote __slotmpryfmjfndhxqkit__ __slotxjuovedfnqfthnsd__ __slotoilrzohvsemqjolq__ sqrt(__slotaafdonjubicrduvj__) end
:__slotpaidoknxlkslmafk__
9
:__slotzslqrmqghxungydo__
1.22284
:__slotjwkhalaszinsfwgl__
5
:__slotbxlefkztewexgeky__
4
quote __slotbxlefkztewexgeky__ __slotjwkhalaszinsfwgl__ __slotpaidoknxlkslmafk__ __slotzslqrmqghxungydo__ end
:__slotmfkrzlgxybxupizv__
1.22284
:__slotmfkrzlgxybxupizv__
SlottedDisplay
We use print
to turn the expression into source code.
For each line, we regex-search for slot variables, and we split the line around those. The code segments around slots are rendered inside <pre-ish>
tags (like <pre>
but inline), and the slots are replaced by embedded displays of the objects.
preish (generic function with 1 method)
SlottedDisplay
"slotted-code {\n\tfont-family: \"JuliaMono\", monospace;\n\tfont-size: .75rem;\n\tdisplay: flex;\n\tflex-direction: column;\n}\npre-ish {\n\twhite-space: pre;\n}\n\nline-like {\n\tdisplay: flex;\n\talign-items: baseline;\n}\n"
plot (generic function with 1 method)
UndefVarError: plot not defined
Here is what happened, the most recent locations are first:
Another cell defining rs contains errors.
Frame viewer
A widget that takes a series of elements and displays them as 'video frames' with a timeline scrubber.
Another cell defining rs contains errors.
frames (generic function with 1 method)
Macro to test frames
@visual_debug (macro with 1 method)
UndefVarError: plot not defined
Here is what happened, the most recent locations are first:
Appendix
DisplayOnly
is_inside_pluto (generic function with 1 method)
@displayonly expression
Marks a expression as Pluto-only, which means that it won't be executed when running outside Pluto. Do not use this for your own projects.
The opposite of @skip_as_script
"hello"
PlutoUI favourites
"@media screen and (min-width: 1081px) {\n\t.plutoui-toc.aside {\n\t\tposition:fixed; \n\t\tright: 1rem;\n\t\ttop: 5rem; \n\t\twidth:25%; \n\t\tpadding: 10px;\n\t\tborder: 3px solid rgba(0, 0, 0, 0.15);\n\t\tborder-radius: 10px;\n\t\tbox-shadow: 0 0 11px 0px #00000010;\n\t\t/* That is, viewport minus top minus Live Docs */\n\t\tma" ⋯ 790 bytes ⋯ " 0px;\n}\n.plutoui-toc.indent section a.H2 {\n\tpadding-left: 10px;\n}\n.plutoui-toc.indent section a.H3 {\n\tpadding-left: 20px;\n}\n.plutoui-toc.indent section a.H4 {\n\tpadding-left: 30px;\n}\n.plutoui-toc.indent section a.H5 {\n\tpadding-left: 40px;\n}\n.plutoui-toc.indent section a.H6 {\n\tpadding-left: 50px;\n}\n"
#1 (generic function with 1 method)
toc (generic function with 1 method)
Slider
Dump (generic function with 1 method)
Base.RefValue{Int64}
x: Int64 1
Show